001 /* 002 * Copyright 2005 Niclas Hedhman 003 * Copyright 2005 Stephen J. McConnell. 004 * 005 * Licensed under the Apache License, Version 2.0 (the "License"); 006 * you may not use this file except in compliance with the License. 007 * You may obtain a copy of the License at 008 * 009 * http://www.apache.org/licenses/LICENSE-2.0 010 * 011 * Unless required by applicable law or agreed to in writing, software 012 * distributed under the License is distributed on an "AS IS" BASIS, 013 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or 014 * implied. 015 * 016 * See the License for the specific language governing permissions and 017 * limitations under the License. 018 */ 019 020 package net.dpml.transit.link; 021 022 import net.dpml.util.StreamUtils; 023 024 import java.io.ByteArrayInputStream; 025 import java.io.ByteArrayOutputStream; 026 import java.io.IOException; 027 import java.io.InputStream; 028 import java.io.OutputStream; 029 import java.net.URI; 030 import java.net.URL; 031 import java.security.AccessController; 032 import java.security.PrivilegedActionException; 033 import java.security.PrivilegedExceptionAction; 034 035 import net.dpml.transit.Artifact; 036 import net.dpml.transit.NullArgumentException; 037 import net.dpml.transit.artifact.ArtifactNotFoundException; 038 import net.dpml.util.PropertyResolver; 039 040 /** 041 * A link manager that maintains persistent link information as a resource. 042 * Link resource located using the [cache]/[group]/[name]/[type]s/[name]-[version].[type].link 043 * resource naming convention. 044 * 045 * Applications should not call the methods for the LinkManager directly, 046 * and it is likely that the LinkManager remains outside the reachability of 047 * applications. 048 * @author <a href="http://www.dpml.net">The Digital Product Meta Library</a> 049 * @version 1.0.0 050 */ 051 public class ArtifactLinkManager 052 implements LinkManager 053 { 054 /** 055 * Sets the URI for the provided Link. 056 * The LinkManager is required to persist this information between 057 * JVM restarts and should be persisted on a scope larger than a 058 * single JVM, typically a host or a local area network. LinkManagers 059 * are encouraged to establish other virtual scopes independent of 060 * network topologies. 061 * 062 * @param linkUri the uri of the link resource 063 * @param targetUri the uri that the link redirects to 064 * @exception IOException if the mapping could not be updated. 065 */ 066 public void setTargetURI( final URI linkUri, final URI targetUri ) 067 throws IOException 068 { 069 if( null == linkUri ) 070 { 071 throw new NullArgumentException( "linkUri" ); 072 } 073 074 try 075 { 076 AccessController.doPrivileged( 077 new PrivilegedExceptionAction() 078 { 079 public Object run() 080 throws IOException 081 { 082 String artifact = linkUri.toASCIIString(); 083 URL store = new URL( null, artifact, new net.dpml.transit.artifact.Handler() ); 084 OutputStream out = store.openConnection().getOutputStream(); 085 byte[] array = getByteArray( targetUri ); 086 ByteArrayInputStream in = new ByteArrayInputStream( array ); 087 StreamUtils.copyStream( in, out, true ); 088 return null; // nothing to return 089 } 090 } 091 ); 092 } 093 catch( PrivilegedActionException e ) 094 { 095 throw (IOException) e.getException(); 096 } 097 } 098 099 /** 100 * Returns the URI that the provided link URI instance is pointing to. 101 * @param linkUri the link uri from which the target will be resolved 102 * @exception LinkNotFoundException if the supplied link uri could not be located 103 * @exception IOException if the mapping could not be retrieved, due to 104 * an IOException during link retrival. 105 * @return target URI that the link points to (possibly null if the link does 106 * not declare a target) 107 */ 108 public URI getTargetURI( final URI linkUri ) 109 throws IOException, LinkNotFoundException 110 { 111 try 112 { 113 URI result = (URI) AccessController.doPrivileged( 114 new PrivilegedExceptionAction() 115 { 116 public Object run() 117 throws IOException 118 { 119 URL store = null; 120 if( Artifact.isRecognized( linkUri ) ) 121 { 122 String artifact = linkUri.toASCIIString(); 123 store = new URL( null, artifact, new net.dpml.transit.artifact.Handler() ); 124 } 125 else 126 { 127 store = linkUri.toURL(); 128 } 129 130 ByteArrayOutputStream out = new ByteArrayOutputStream(); 131 InputStream in = store.openConnection().getInputStream(); 132 StreamUtils.copyStream( in, out, true ); 133 String target = out.toString( "ISO8859-1" ); 134 String path = PropertyResolver.resolve( target ); 135 URI value = URI.create( path ); 136 return value; 137 } 138 } 139 ); 140 return result; 141 } 142 catch( PrivilegedActionException e ) 143 { 144 Exception exception = e.getException(); 145 if( exception instanceof ArtifactNotFoundException ) 146 { 147 final String error = 148 "Link not found: " 149 + linkUri; 150 throw new LinkNotFoundException( error, linkUri ); 151 } 152 else 153 { 154 throw (IOException) exception; 155 } 156 } 157 } 158 159 private byte[] getByteArray( URI uri ) throws IOException 160 { 161 if( null != uri ) 162 { 163 return uri.toString().getBytes( "ISO8859-1" ); 164 } 165 else 166 { 167 return new byte[0]; 168 } 169 } 170 }